home *** CD-ROM | disk | FTP | other *** search
/ MacFormat España 26 / macformat_26.iso / Shareware / Programación / C Reference Card / C Reference Card.rsrc / TEXT_406_6.txt < prev    next >
Text File  |  1997-01-29  |  6KB  |  194 lines

  1. POINTERS : using pointers, arrays, aliases, handles, function pointers
  2. _________________________________________________________________________
  3.  
  4.  
  5. USING POINTERS
  6.  
  7. Pointers are indispensable and are frequently used in C programs, but they tend to be one of the stumbling blocks for new programmers and can crash a program in a heartbeat if misused.  The concept of a pointer is simple; a pointer is a variable which contains the computer address of something in memory.  You define a pointer as follows:
  8.  
  9.   int *myPtr;     // define a pointer to an integer variable type.
  10.  
  11. Another method of defining a pointer looks like this:
  12.  
  13.   int* myPtr;     // another way to define a pointer.
  14.  
  15. You assign values to pointers, and dereference pointers as follows:
  16.  
  17.   int i = 5;      // declare integer and assign value of 5.
  18.   int j;          // declare integer.
  19.   int *ip;        // declare pointer to type int.
  20.  
  21.   ip = &i;        // address of 'i' stored in variable 'ip'.
  22.   j = *ip;        // dereference pointer to return value at address.
  23.   cout << i << j; // value of both 'i' and 'j' are the same.
  24.  
  25. Pointers are a necessary evil in C programming.  They provide you with a lot of power, but one bad pointer can ruin your day.  If you pass a bad pointer to a Toolbox function like PaintRect() your program will likely crash.  Here's a simple snippet which shows the basics of declaring a pointer and assigning it the address of a variable:
  26.  
  27.   // pointer.cp
  28.   #include <iostream.h>
  29.  
  30.   void myPrintAddr( int j );  // pass by value
  31.  
  32.   void main( void )
  33.   {
  34.     int i = 1;
  35.     int *iPtr;
  36.  
  37.     iPtr = &i;                // &i returns address of 'i'
  38.     cout << "The value of i is " << i << endl;
  39.     cout << "The address of i is " << iPtr << endl;
  40.     cout << "The value at address " << iPtr << " is " 
  41.       << *iPtr << endl;       // *iPtr 'dereferences' pointer iPtr
  42.     myPrintAddr( i );
  43.   }
  44.  
  45.   void myPrintAddr( int j )
  46.   {
  47.     int *jPtr;
  48.   
  49.     jPtr = &j;
  50.     cout << "The value of j is " << j << endl;
  51.     cout << "The address of j is " << jPtr << endl;
  52.     cout << "Value's the same, address is different!" << endl;
  53.   }
  54.   // end pointer.cp
  55.  
  56. When incrementing pointers, the pointer is incremented by the size of the object it points to.  For example, if myPtr points to a data structure which has a size of X bytes, myPtr++ will add X to the variable myPtr.
  57.  
  58.  
  59.  
  60. ARRAYS
  61.  
  62. In C, an array is a pointer to a sequential block of related data types.  For example, "float myArray[10]" would define a pointer "myArray" to the beginning of a block of memory sufficient to hold 10 variables of type float.  The first item in an array is accessed with a "0" subscript.  Therefore, "int i[5]" defines an array of 5 integers, i[0], i[1], i[2], i[3], and i[4].  Here's an example showing the use of arrays:
  63.  
  64.   // array.cp
  65.   #include <iostream.h>
  66.  
  67.   #define kSize 5
  68.  
  69.   void main( void )
  70.   {
  71.     int   one[kSize] = {2,4,6,8,10};
  72.     int   two[2][kSize] = { {1,2,3,4,5},{11,12,13,14,15} };
  73.  
  74.     cout << "Single Dimensional Array" << endl;
  75.     for( int i = 0; i<kSize; i++ )
  76.     {
  77.       cout << i << " = " << one[i] << endl;
  78.     }
  79.  
  80.     cout << endl << "Multiple Dimensional Array" << endl;
  81.     for( i = 0; i<kSize; i++ )
  82.     {
  83.       cout << "0," << i << " = " << two[0][i] << endl;
  84.       cout << "1," << i << " = " << two[1][i] << endl;
  85.     }
  86.   }
  87.   // end array.cp
  88.  
  89.  
  90.  
  91. ALIASES
  92.  
  93. C++ allows you to create aliases (also known as references).  As the name implies, an alias is just another name for the same item.  You can use aliases to pass function variables by reference rather than by value.  This can alleviate a lot of headaches.  If you find yourself having to dereference pointer variables, try using aliases instead.
  94.  
  95.   // alias.cp
  96.   /*
  97.      Compare results to "pointer.cp" snippet above.
  98.   */
  99.  
  100.   #include <iostream.h>
  101.  
  102.   void myPrintAddr( int &j );  // pass by reference
  103.  
  104.   void main( void )
  105.   {
  106.     int i = 1;
  107.     int *iPtr;
  108.  
  109.     iPtr = &i;
  110.     cout << "The value of i is " << i << endl;
  111.     cout << "The address of i is " << iPtr << endl;
  112.     myPrintAddr( i );
  113.     cout << "They match!" << endl;
  114.   }
  115.  
  116.   void myPrintAddr( int &j )
  117.   {
  118.     int *jPtr;
  119.  
  120.     jPtr = &j;
  121.     cout << "The value of j is " << j << endl;
  122.     cout << "The address of j is " << jPtr << endl;
  123.   }
  124.   // end alias.cp
  125.  
  126.  
  127.  
  128. MACINTOSH HANDLES
  129.  
  130. The MacOS makes use of handles.  Like pointers, they're not difficult to understand, but they can be the source of headaches when your debugging code.  Handles are, simply, pointers to pointers.
  131.  
  132.   int myInt;                    // declares integer.
  133.   int *myIntPtr;                // declares pointer.
  134.   int **myIntH;                 // declares handle.
  135.  
  136. You normally don't declare handles as shown above.  You typically use standard MacOS definitions like:
  137.  
  138.   Handle myH;                   // generic MacOS handle.
  139.   TEHandle myTE;                // handle to MacOS TE data structure.
  140.  
  141. Then use Toolbox functions that return pointers to "master pointers" like:
  142.  
  143.   myH = NewHandle( 1024 );      // returns handle to 1024-sized block.
  144.   myTE = NewTE( dRect, vRect ); // returns handle to TE structure.
  145.  
  146. Why handles?  That's easy - memory fragmentation.  The MacOS is constantly on the look for items which it can move to fill-in memory gaps.  Handles allow the operating system to do just that.  Should an item be moved, the value of the master pointer is updated, but the value of the handle remains the same.
  147.  
  148.  
  149.  
  150. POINTERS TO FUNCTIONS
  151.  
  152. You can define pointers to functions which can be used in arrays, returned by functions, or used to call the associated functions themselves.
  153.  
  154.   // complex.cp
  155.   #include <iostream.h>
  156.   #include <stdlib.h>
  157.  
  158.   int myCallFunction( int (*function)( int ), int mySeed );
  159.   int myRandom( int theSeed );
  160.  
  161.   void main( void )
  162.   {
  163.     int mySeed = 1;
  164.     int result;
  165.  
  166.     while( mySeed )
  167.     {
  168.       cout << "Enter seed (zero to exit): ";
  169.       cin >> mySeed;
  170.       if( mySeed )
  171.       {
  172.         result = myCallFunction( myRandom, mySeed );
  173.         cout << "Result: " << result << endl;
  174.       }
  175.     }
  176.   }
  177.  
  178.   int myCallFunction( int (*function)( int ), int input )
  179.   {
  180.     int random;
  181.  
  182.     random = function( input );
  183.     return random;
  184.   }
  185.  
  186.   int myRandom( int theSeed )
  187.   {
  188.     int result;
  189.  
  190.     srand( theSeed );
  191.     result = rand();
  192.     return result;
  193.   }
  194.   // end complex.cp